# 一、客户端渲染与服务端渲染
# 1.1 什么是客户端渲染
react在客户端执行,消耗客户端性能。客户端渲染,页面初始加载的
HTML页面中无网页展示内容,需要加载执行JavaScript文件中的React代码,通过JavaScript渲染生成页面,同时,JavaScript代码会完成页面交互事件的绑定,详细流程可参考下图
客户端渲染流程
浏览器发送请求-->服务器返回HTML-->浏览器发送bundle.js请求-->服务器返回bundle.js-->浏览器执行bundle.js中的react代码完成渲染

# 1.2 什么是服务端渲染
react在服务端执行,消耗服务端性能。我们所说的服务端渲染,只是首次加载页面的时候,后面都是通过前端路由进行客户端渲染
- 用户请求服务器,服务器上直接生成 
HTML内容并返回给浏览器。服务器端渲染来,页面的内容是由Server端生成的。一般来说,服务器端渲染的页面交互能力有限,如果要实现复杂交互,还是要通过引入JavaScript文件来辅助实现 
服务端渲染流程
浏览器发送请求-->服务器运行React代码生成页面-->服务器返回页面
# 1.3 什么是同构
一套react代码,在服务端执行一次,在客户端也执行一次。在服务端执行同构
renderToString只是返回界面展示,并不能绑定事件,需要在客户端再次执行js代码绑定事件
服务器运行React代码渲染出HTML-->发送HTML给浏览器-->浏览器接收到内容展示-->浏览器加载js文件-->Js中的React代码在浏览器端重新执行-->JS中的React代码接管页面操作
路由同构
让路由在服务端、客户端各跑一遍
# 1.4 使用SSR优劣势
一般情况下,当我们使用
React编写代码时,页面都是由客户端执行JavaScript逻辑动态挂DOM生成的,也就是说这种普通的单页面应用实际上采用的是客户端渲染模式。在大多数情况下,客户端渲染完全能够满足我们的业务需求,那为什么我们还需要SSR这种同构技术呢
CSR项目的TTFP(Time To First Page)时间比较长,参考之前的图例,在CSR的页面渲染流程中,首先要加载HTML文件,之后要下载页面所需的JavaScript文件,然后JavaScript文件渲染生成页面。在这个渲染过程中至少涉及到两个HTTP请求周期,所以会有一定的耗时,这也是为什么大家在低网速下访问普通的React或者Vue应用时,初始页面会有出现白屏的原因CSR项目的SEO能力极弱,在搜索引擎中基本上不可能有好的排名。因为目前大多数搜索引擎主要识别的内容还是HTML,对JavaScript文件内容的识别都还比较弱。如果一个项目的流量入口来自于搜索引擎,这个时候你使用 CSR 进行开发,就非常不合适了
SSR的产生,主要就是为了解决上面所说的两个问题。在React中使用 SSR 技术,我们让 React 代码在服务器端先执行一次,使得用户下载的 HTML 已经包含了所有的页面展示内容,这样,页面展示的过程只需要经历一个HTTP请求周期,TTFP时间得到一倍以上的缩减
- 同时,由于 
HTML中已经包含了网页的所有内容,所以网页的SEO效果也会变的非常好。之后,我们让React代码在客户端再次执行,为HTML网页中的内容添加数据及事件的绑定,页面就具备了React的各种交互能力 
但是,
SSR这种理念的实现,并非易事。我们来看一下在React中实现SSR技术的架构图:

- 使用 
SSR这种技术,将使原本简单的React项目变得非常复杂,项目的可维护性会降低,代码问题的追溯也会变得困难 - 所以,使用 
SSR在解决问题的同时,也会带来非常多的副作用,有的时候,这些副作用的伤害比起SSR技术带来的优势要大的多。从个人经验上来说,我一般建议大家,除非你的项目特别依赖搜索引擎流量,或者对首屏时间有特殊的要求,否则不建议使用SSR 
# 二、SSR的实现本质
SSR 之所以能够实现,本质上是因为虚拟 DOM 的存在
SSR的工程中,React代码会在客户端和服务器端各执行一次。你可能会想,这没什么问题,都是 JavaScript 代码,既可以在浏览器上运行,又可以在Node环境下运行。但事实并非如此,如果你的React代码里,存在直接操作DOM的代码,那么就无法实现SSR这种技术了,因为在Node环境下,是没有DOM这个概念存在的,所以这些代码在Node环境下是会报错的- 在 
React框架中引入了一个概念叫做虚拟 DOM,虚拟 DOM 是真实 DOM 的一个 JavaScript 对象映射,React 在做页面操作时,实际上不是直接操作 DOM,而是操作虚拟 DOM,也就是操作普通的 JavaScript 对象,这就使得 SSR 成为了可能。在服务器,我可以操作 JavaScript 对象,判断环境是服务器环境,我们把虚拟DOM映射成字符串输出;在客户端,我也可以操作 JavaScript 对象,判断环境是客户端环境,我就直接将虚拟 DOM 映射成真实 DOM,完成页面挂载 
# 三、SSR中服务器端路由和客户端路由区别
实现
React的SSR架构,我们需要让相同的React代码在客户端和服务器端各执行一次。大家注意,这里说的相同的React代码,指的是我们写的各种组件代码,所以在同构中,只有组件的代码是可以公用的,而路由这样的代码是没有办法公用的,大家思考下这是为什么呢?其实原因很简单,在服务器端需要通过请求路径,找到路由组件,而在客户端需通过浏览器中的网址,找到路由组件,是完全不同的两套机制,所以这部分代码是肯定无法公用。我们来看看在 SSR 中,前后端路由的实现代码
# 3.1 客户端路由
const App = () => {
  return (
    <Provider store={store}>
      <BrowserRouter>
        <div>
          <Route path='/' component={Home}>
  		</div>
      </BrowserRouter>
    </Provider>
  )
}
ReactDom.render(<App/>, document.querySelector('#root'))
客户端路由代码非常简单,大家一定很熟悉,
BrowserRouter会自动从浏览器地址中,匹配对应的路由组件显示出来
# 3.2 服务器端路由
const App = () => {
  return 
    <Provider store={store}>
      <StaticRouter location={req.path} context={context}>
        <div>
          <Route path='/' component={Home}>
        </div>
      </StaticRouter>
    </Provider>
}
Return ReactDom.renderToString(<App/>)
服务器端路由代码相对要复杂一点,需要你把
location(当前请求路径)传递给StaticRouter组件,这样StaticRouter才能根据路径分析出当前所需要的组件是谁。(PS:StaticRouter是React-Router针对服务器端渲染专门提供的一个路由组件。)
- 通过 
BrowserRouter我们能够匹配到浏览器即将显示的路由组件,对浏览器来说,我们需要把组件转化成DOM,所以需要我们使用ReactDom.render方法来进行DOM的挂载。而StaticRouter能够在服务器端匹配到将要显示的组件,对服务器端来说,我们要把组件转化成字符串,这时我们只需要调用ReactDom提供的renderToString方法,就可以得到App组件对应的HTML字符串。 - 对于一个 
React应用来说,路由一般是整个程序的执行入口。在SSR中,服务器端的路由和客户端的路由不一样,也就意味着服务器端的入口代码和客户端的入口代码是不同的 - 我们知道, 
React代码是要通过Webpack打包之后才能运行的,实际上是源代码打包过后生成的代码。上面也说到,服务器端和客户端渲染中的代码,只有一部分一致,其余是有区别的。所以,针对代码运行环境的不同,要进行有区别的Webpack打包 
# 四、服务端和客户端打包配置
# 4.1 客户端 Webpack 配置
{
  entry: './src/client/index.js',
  output: {
    filename: 'index.js',
    path: path.resolve(__dirname, 'public')
  },
  module: {
    rules: [{
      test: /\.js?$/,
      loader: 'babel-loader'
    },{
      test: /\.css?$/,
      use: ['style-loader', {
        loader: 'css-loader',
        options: {modules: true}
      }]
    },{
      test: /\.(png|jpeg|jpg|gif|svg)?$/,
      loader: 'url-loader',
      options: {
        limit: 8000,
        publicPath: '/'
      }
    }]
  }
}
# 4.2 服务器端 Webpack 配置
{
  target: 'node',
  entry: './src/server/index.js',
  output: {
    filename: 'bundle.js',
    path: path.resolve(__dirname, 'build')
  },
  externals: [nodeExternals()],
  module: {
    rules: [{
      test: /\.js?$/,
      loader: 'babel-loader'
    },{
      test: /\.css?$/,
      use: ['isomorphic-style-loader', {
        loader: 'css-loader',
        options: {modules: true}
      }]
    },{
      test: /\.(png|jpeg|jpg|gif|svg)?$/,
      loader: 'url-loader',
      options: {
        limit: 8000,
        outputPath: '../public/',
        publicPath: '/'
      }
    }]
  }
};
- 在 
SSR中,服务器端渲染的代码和客户端的代码的入口路由代码是有差异的,所以在Webpack中,Entry的配置首先肯定是不同的 - 在服务器端运行的代码,有时我们需要引入 
Node中的一些核心模块,我们需要Webpack做打包的时候能够识别出类似的核心模块,一旦发现是核心模块,不必把模块的代码合并到最终生成的代码中,解决这个问题的方法非常简单,在服务器端的Webpack配置中,你只要加入target: node这个配置即可 - 服务器端渲染的代码,如果加载第三方模块,这些第三方模块也是不需要被打包到最终的源码中的,因为 
Node环境下通过NPM已经安装了这些包,直接引用就可以,不需要额外再打包到代码里。为了解决这个问题,我们可以使用webpack-node-externals这个插件,代码中的nodeExternals指的就是这个插件,通过这个插件,我们就能解决这个问题。关于Node这里的打包问题,可能看起来有些抽象,不是很明白的同学可以仔细读一下webpack-node-externals相关的文章或文档,你就能很好的明白这里存在的问题了 - 当我们的 
React代码中引入了一些CSS样式代码时,服务器端打包的过程会处理一遍CSS,而客户端又会处理一遍。查看配置,我们可以看到,服务器端打包时我们用了isomorphic-style-loader,它处理CSS的时候,只在对应的DOM元素上生成class类名,然后返回生成的CSS样式代码 - 而在客户端代码打包配置中,我们使用了 
css-loader和style-loader,css-loader不但会在DOM上生成class类名,解析好的CSS代码,还会通过style-loader把代码挂载到页面上。不过这么做,由于页面上的样式实际上最终是由客户端渲染时添加上的,所以页面可能会存在一开始没有样式的情况,为了解决这个问题, 我们可以在服务器端渲染时,拿到isomorphic-style-loader返回的样式代码,然后以字符串的形式添加到服务器端渲染的HTML之中 - 而对于图片等类型的文件引入,
url-loader也会在服务器端代码和客户端代码打包的过程中分别进行打包,这里,我偷了一个懒,无论服务器端打包还是客户端打包,我都让打包生成的文件存储在public目录下,这样,虽然文件会打包出来两遍,但是后打包出来的文件会覆盖之前的文件,所以看起来还是只有一份文件 - 如果想进行优化,你可以让图片的打包只进行一次,借助一些 
Webpack的插件,实现这个也并非难事,你甚至可以自己也写一个loader,来解决这样的问题 - 如果你的 
React应用中没有异步数据的获取,单纯的做一些静态内容展示,经过上面的配置,你会发现一个简单的SSR应用很快的就可以被实现出来了。但是,真正的一个React项目中,我们肯定要有异步数据的获取,绝大多数情况下,我们还要使用Redux管理数据。而如果想在SSR应用中实现,就不是这么简单 
# 五、SSR 中异步数据的获取 + Redux 的使用
客户端渲染中,异步数据结合 Redux 的使用方式遵循下面的流程
- 创建 
Store - 根据路由显示组件
 - 派发 
Action获取数据 - 更新 
Store中的数据 - 组件 
Rerender 
而在服务器端,页面一旦确定内容,就没有办法 Rerender 了,这就要求组件显示的时候,就要把 Store 的数据都准备好,所以服务器端异步数据结合 Redux 的使用方式
服务器端异步数据结合 Redux流程
- 创建 
Store - 根据路由分析 
Store中需要的数据 - 派发 
Action获取数据 - 更新
Store中的数据 - 结合数据和组件生成 
HTML,一次性返回 
# 5.1 服务器端渲染的流程
创建
Store:这一部分有坑,要注意避免,大家知道,客户端渲染中,用户的浏览器中永远只存在一个Store,所以代码上你可以这么写
const store = createStore(reducer, defaultState)
export default store;
然而在服务器端,这么写就有问题了,因为服务器端的
Store是所有用户都要用的,如果像上面这样构建Store,Store变成了一个单例,所有用户共享Store,显然就有问题了。所以在服务器端渲染中,Store的创建应该像下面这样,返回一个函数,每个用户访问的时候,这个函数重新执行,为每个用户提供一个独立的Store
const getStore = (req) => {
  return createStore(reducer, defaultState);
}
export default getStore;
- 根据路由分析 Store 中需要的数据: 要想实现这个步骤,在服务器端,首先我们要分析当前出路由要加载的所有组件,这个时候我们可以借助一些第三方的包,比如说 react-router-config, 具体这个包怎么使用,不做过多说明,大家可以查看文档,使用这个包,传入服务器请求路径,它就会帮助你分析出这个路径下要展示的所有组件
 - 派发 Action 获取数据: 接下来,我们在每个组件上增加一个获取数据的方法
 
Home.loadData = (store) => {
  return store.dispatch(getHomeList())
}
这个方法需要你把服务器端渲染的 Store 传递进来,它的作用就是帮助服务器端的 Store 获取到这个组件所需的数据。 所以,组件上有了这样的方法,同时我们也有当前路由所需要的所有组件,依次调用各个组件上的 loadData 方法,就能够获取到路由所需的所有数据内容了
更新 Store 中的数据: 其实,当我们执行第三步的时候,已经在更新 Store 中的数据了,但是,我们要在生成 HTML 之前,保证所有的数据都获取完毕,这怎么处理呢
// matchedRoutes 是当前路由对应的所有需要显示的组件集合
matchedRoutes.forEach(item => {
  if (item.route.loadData) {
    const promise = new Promise((resolve, reject) => {
      item.route.loadData(store).then(resolve).catch(resolve);
    })
    promises.push(promise);
  }
})
Promise.all(promises).then(() => {
  // 生成 HTML 逻辑
})
- 这里,我们使用 
Promise来解决这个问题,我们构建一个Promise队列,等待所有的Promise都执行结束后,也就是所有store.dispatch都执行完毕后,再去生成HTML。这样的话,我们就实现了结合Redux的SSR流程 - 在上面,我们说到,服务器端渲染时,页面的数据是通过 
loadData函数来获取的。而在客户端,数据获取依然要做,因为如果这个页面是你访问的第一个页面,那么你看到的内容是服务器端渲染出来的,但是如果经过react-router路由跳转道第二个页面,那么这个页面就完全是客户端渲染出来的了,所以客户端也要去拿数据 - 在客户端获取数据,使用的是我们最习惯的方式,通过 
componentDidMount进行数据的获取。这里要注意的是,componentDidMount只在客户端才会执行,在服务器端这个生命周期函数是不会执行的。所以我们不必担心componentDidMount和loadData会有冲突,放心使用即可。这也是为什么数据的获取应该放到componentDidMount这个生命周期函数中而不是componentWillMount中的原因,可以避免服务器端获取数据和客户端获取数据的冲突 
# 六、处理styled-components样式
styled-components暴露了ServerStyleSheet,它允许我们用 中的所有styled组件创建一个样式表。这个样式表稍后会传入我们的Html模板
//render.js
import React from 'react';
import {StaticRouter} from 'react-router-dom'
import {renderToString} from 'react-dom/server'
import renderHTML from './template'
import { Provider } from 'react-redux'
import { ServerStyleSheet } from 'styled-components';
export const render = (store,routes,req,context)=>{
    const sheet = new ServerStyleSheet(); // <-- 创建样式表
    const content = renderToString(
      // 收集样式
      sheet.collectStyles(
        <Provider store={store}>
          <StaticRouter location={req.path} context={context}>
            {routes}
          </StaticRouter>
        </Provider>
      )
    )
    const styles = sheet.getStyleTags(); // <-- 从表中获取所有标签
    return renderHTML(content,store,styles)
}
将
styles作为参数添加到我们的Html函数中,并将$ {styles}参数插入到我们的模板字符串中
// template.js
export default (content,store,styles)=>`
  <!DOCTYPE html>
    <html lang="en">
      <head>
        <meta charset="utf-8">
        <meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1, minimum-scale=1, user-scalable=no" />
        <meta name="theme-color" content="#000000">
        <meta name="apple-mobile-web-app-capable" content="yes">
        <meta name="apple-touch-fullscreen" content="yes">
        <title>好物为您聚集大平台的优惠商品,让你更便捷的找到你想要的宝物</title>
        <link rel="shortcut icon" href="/favicon.ico">
        <link rel="stylesheet" href="${buildPath['vendor.css']}" />
        <link rel="stylesheet" href="${buildPath['main.css']}" />
        ${styles}
      </head>
      <body>
      <div id="root">${content}</div>
      <script>
        window.context = {
          state: ${JSON.stringify(store.getState())}
        }
      </script>
      </body>
  </html>
`
# 七、Node 只是一个中间层
在
SSR架构中,一般 Node 只是一个中间层,用来做React代码的服务器端渲染,而Node需要的数据通常由 API 服务器单独提供

服务器端渲染时,直接请求
API服务器的接口获取数据没有任何问题。但是在客户端,就有可能存在跨域的问题了,所以,这个时候,我们需要在服务器端搭建Proxy代理功能,客户端不直接请求API服务器,而是请求Node服务器,经过代理转发,拿到API服务器的数据
- 这里你可以通过 
express-http-proxy这样的工具帮助你快速搭建Proxy代理功能,但是记得配置的时候,要让代理服务器不仅仅帮你转发请求,还要把cookie携带上,这样才不会有权限校验上的一些问题。 
// Node 代理功能实现代码
app.use('/api', proxy('http://apiServer.com', {
  proxyReqPathResolver: function (req) {
    return '/ssr' + req.url;
  }
}));
# 八、完整代码示例
https://github.com/poetries/react-ssr